Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix required tenant in OpenId Validation settings UI #17266

Merged
merged 3 commits into from
Dec 24, 2024

Conversation

salmattia
Copy link
Contributor

Fix #17233. Now one of tenant or authority is required. Tenant can be empty when an external authority validates tokens.
Error is displayed on save only if both tenant and authority fields are not filled out

…r, if authority is provided, tenant is not required
Copy link
Contributor

Thank you for submitting your first pull request, awesome! 🚀 If you haven't already, please take a moment to review our contribution guide. This guide provides helpful information to ensure your contribution aligns with our standards. A core team member will review your pull request.

@salmattia
Copy link
Contributor Author

@dotnet-policy-service agree company="Imatra"

Copy link
Member

@kevinchalet kevinchalet left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for your PR, @salmattia!

@@ -72,7 +72,10 @@ public override async Task<IDisplayResult> UpdateAsync(OpenIdValidationSettings

if (string.IsNullOrWhiteSpace(model.Tenant))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While we're at it, let's use string.IsNullOrEmpty() instead of string.IsNullOrWhiteSpace() lines 73 and 75 😃

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done!

Copy link
Member

@kevinchalet kevinchalet left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks great!

I'll let @MikeAlhayek merge it and backport it to make sure it's included in the next patch/minor release.

Happy holidays! 🎄

@salmattia
Copy link
Contributor Author

Looks great!

I'll let @MikeAlhayek merge it and backport it to make sure it's included in the next patch/minor release.

Happy holidays! 🎄

Thank you! Happy holidays

@MikeAlhayek
Copy link
Member

MikeAlhayek commented Dec 20, 2024

@kevinchalet I see couple of issues with the code. I think the code should look like this instead. do you agree?

I think the old code should be kept. Instead, replace the line

settings.Authority = !string.IsNullOrEmpty(model.Authority) ? new Uri(model.Authority, UriKind.Absolute) : null;

with this code

        if (string.IsNullOrEmpty(model.Authority))
        {
            context.Updater.ModelState.AddModelError(Prefix, nameof(model.Authority), S["A tenant or authority value is required."]);
        }
        else 
        {
            settings.Authority = new Uri(model.Authority, UriKind.Absolute)
        }

If Authority should be only validated if there is not Tenant, then we may need to fix the error to this instead

context.Updater.ModelState.AddModelError(Prefix, nameof(model.Authority), S["A tenant or authority value is required."]);

@kevinchalet
Copy link
Member

I think the code should look like this instead. do you agree?

Personally, I'd even remove all this validation code and simply rely on OpenIdValidationService, that eventually performs similar checks anyway:

public async Task<ImmutableArray<ValidationResult>> ValidateSettingsAsync(OpenIdValidationSettings settings)
{
ArgumentNullException.ThrowIfNull(settings);
var results = ImmutableArray.CreateBuilder<ValidationResult>();
if (!(settings.Authority == null ^ string.IsNullOrEmpty(settings.Tenant)))
{
results.Add(new ValidationResult(S["Either a tenant or an authority must be registered."], new[]
{
nameof(settings.Authority),
nameof(settings.Tenant),
}));
}
if (settings.Authority != null)
{
if (!settings.Authority.IsAbsoluteUri || !settings.Authority.IsWellFormedOriginalString())
{
results.Add(new ValidationResult(S["The specified authority is not valid."], new[]
{
nameof(settings.Authority)
}));
}
if (!string.IsNullOrEmpty(settings.Authority.Query) || !string.IsNullOrEmpty(settings.Authority.Fragment))
{
results.Add(new ValidationResult(S["The authority cannot contain a query string or a fragment."], new[]
{
nameof(settings.Authority),
}));
}
}
if (settings.MetadataAddress != null)
{
if (!settings.MetadataAddress.IsAbsoluteUri || !settings.MetadataAddress.IsWellFormedOriginalString())
{
results.Add(new ValidationResult(S["The specified metadata address is not valid."], new[]
{
nameof(settings.MetadataAddress),
}));
}
if (!string.IsNullOrEmpty(settings.MetadataAddress.Query) || !string.IsNullOrEmpty(settings.MetadataAddress.Fragment))
{
results.Add(new ValidationResult(S["The metadata address cannot contain a query string or a fragment."], new[]
{
nameof(settings.MetadataAddress),
}));
}
if (!string.IsNullOrEmpty(settings.Tenant))
{
results.Add(new ValidationResult(S["No metatada address can be set when using another tenant."], new[]
{
nameof(settings.MetadataAddress),
}));
}
}
if (!string.IsNullOrEmpty(settings.Tenant) && !string.IsNullOrEmpty(settings.Audience))
{
results.Add(new ValidationResult(S["No audience can be set when using another tenant."], new[]
{
nameof(settings.Audience),
}));
}
if (settings.Authority != null && string.IsNullOrEmpty(settings.Audience))
{
results.Add(new ValidationResult(S["An audience must be set when configuring the authority."], new[]
{
nameof(settings.Audience),
}));
}
if (settings.Authority == null && settings.DisableTokenTypeValidation)
{
results.Add(new ValidationResult(S["Token type validation can only be disabled for remote servers."], new[]
{
nameof(settings.DisableTokenTypeValidation),
}));
}
if (!string.IsNullOrEmpty(settings.Audience) &&
settings.Audience.StartsWith(OpenIdConstants.Prefixes.Tenant, StringComparison.OrdinalIgnoreCase))
{
results.Add(new ValidationResult(S["The audience cannot start with the special 'oct:' prefix."], new[]
{
nameof(settings.Audience),
}));
}
// If a tenant was specified, ensure it is valid, that the OpenID server feature
// was enabled and that at least a scope linked with the current tenant exists.
if (!string.IsNullOrEmpty(settings.Tenant) &&
!string.Equals(settings.Tenant, _shellSettings.Name, StringComparison.Ordinal))
{
if (!_shellHost.TryGetSettings(settings.Tenant, out var shellSettings))
{
results.Add(new ValidationResult(S["The specified tenant is not valid."]));
}
else
{
var shellScope = await _shellHost.GetScopeAsync(shellSettings);
await shellScope.UsingAsync(async scope =>
{
var options = scope.ServiceProvider.GetRequiredService<IOptionsMonitor<OpenIddictServerOptions>>().CurrentValue;
if (options.UseReferenceAccessTokens)
{
results.Add(new ValidationResult(S["Selecting a server tenant for which reference access tokens are enabled is currently not supported."], new[]
{
nameof(settings.Tenant),
}));
}
var manager = scope.ServiceProvider.GetService<IOpenIdScopeManager>();
if (manager == null)
{
results.Add(new ValidationResult(S["The specified tenant is not valid."], new[]
{
nameof(settings.Tenant),
}));
}
else
{
var resource = OpenIdConstants.Prefixes.Tenant + _shellSettings.Name;
if (!await manager.FindByResourceAsync(resource).AnyAsync())
{
results.Add(new ValidationResult(S["No appropriate scope was found."], new[]
{
nameof(settings.Tenant),
}));
}
}
});
}
}
return results.ToImmutable();
}

@MikeAlhayek
Copy link
Member

@kevinchalet The OpenIdValidationService is good. But in the driver, we assign prefix for the validation error key names. Also, if for some reason the property names in OpenIdValidationSettingsViewModel did not match the same name, some errors may not be printed on the UI.

I think its better to leave the validation in the driver "I know it repeated logic", but it's clear what we are validating and what keys we are settings with the errors.

Please check out the updated validation logic. If you are good with this, I'll merge and portback.

@kevinchalet
Copy link
Member

Also, if for some reason the property names in OpenIdValidationSettingsViewModel did not match the same name, some errors may not be printed on the UI.

Well, if you're concerned by that, we should be duplicating all the validation checks and not just a single one? 😄

If you strongly think we should have a duplicated check, let's merge it. Otherwise, just remove it.

@MikeAlhayek
Copy link
Member

I'll merge this one and back port it. We can submit another PR in v3 that removed the validation from the driver.

@MikeAlhayek MikeAlhayek merged commit d8168f2 into OrchardCMS:main Dec 24, 2024
6 checks passed
Copy link
Contributor

Congratulations on your first PR merge! 🎉 Thank you for your contribution! We're looking forward to welcoming other contributions of yours in the future. @all-contributors please add @salmattia for code.

Copy link
Contributor

@github-actions[bot]

I've put up a pull request to add @salmattia! 🎉

@MikeAlhayek
Copy link
Member

/backport to release/2.1

Copy link
Contributor

Started backporting to release/2.1: https://github.com/OrchardCMS/OrchardCore/actions/runs/12486787988

@MarGraz
Copy link
Contributor

MarGraz commented Jan 10, 2025

Hi @salmattia, @kevinchalet, @MikeAlhayek, I am still experiencing this issue in Orchard Core 2.1.3. The Authorization Server tenant is mandatory, making it impossible to save an external authority.

Are you encountering the same problem in this version? How can I apply the fix?

Thank you

image

P.S.:

Now I fixed with a workaround, modifying the value of OpenIdValidationSettings in the DB, in the Document table, row Type OrchardCore.Settings.SiteSettings, OrchardCore.Settings, from this:

"OpenIdValidationSettings":{"DisableTokenTypeValidation":false,"Tenant":"Default"}}

to this:

"OpenIdValidationSettings":{"Audience":"Store.SSO","Authority":"http://host.docker.internal:12150/","DisableTokenTypeValidation":false,"Tenant":null,"MetadataAddress":null}}

@MikeAlhayek
Copy link
Member

We did not release a new version of OC that includes this fix. This will be part of OC 3.0 and 2.1.4 if we decided we will release a new version.

@kevinchalet
Copy link
Member

This will be part of OC 3.0 and 2.1.4 if we decided we will release a new version.

Since at least 2 people are affected by this bug, we should definitely backport it. Bumping the OpenIddict dependency to 6.0.0 as part of a new minor release would also be nice (otherwise folks will stay in an unsupported state until OC 3.0 ships).

@MikeAlhayek
Copy link
Member

It's already backported just need a release.

Why bump up OpenIdDict library at the same time, I don't want that to cause unexpected issue for others since that is a minor

@kevinchalet
Copy link
Member

kevinchalet commented Jan 11, 2025

It's already backported just need a release.

By backport it, I mean create a branch that includes this change (already done) AND release a new version with the fix. Having a branch and no release is totally useless for those impacted by this bug 😄

Why bump up OpenIdDict library at the same time,

As I said, the 5.x version is no longer supported. 3.0 has no date assigned, so it could ship in a few months... a period during which users can't migrate their OC app to use OpenIddict 6.0, which is the supported version.

@MikeAlhayek
Copy link
Member

Yes I understand. It's back ported just in case we want to release another patch. I can work on a new patch next week if I have time.

However, I am not sure upgrading OpenIdDict in a patch release is a good idea as it could break people. @sebastienros thought on upgrading OpenIdDict to 6.0 in 2.1.4?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

OpenId.Validation feature configuration does not allow authorities from Settings UI
4 participants